1 // Tencent is pleased to support the open source community by making RapidJSON available.
3 // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
5 // Licensed under the MIT License (the "License"); you may not use this file except
6 // in compliance with the License. You may obtain a copy of the License at
8 // http://opensource.org/licenses/MIT
10 // Unless required by applicable law or agreed to in writing, software distributed
11 // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 // specific language governing permissions and limitations under the License.
15 #ifndef RAPIDJSON_ISTREAMWRAPPER_H_
16 #define RAPIDJSON_ISTREAMWRAPPER_H_
24 RAPIDJSON_DIAG_OFF(padded)
25 #elif defined(_MSC_VER)
27 RAPIDJSON_DIAG_OFF(4351) // new behavior: elements of array 'array' will be default initialized
30 RAPIDJSON_NAMESPACE_BEGIN
32 //! Wrapper of \c std::basic_istream into RapidJSON's Stream concept.
34 The classes can be wrapped including but not limited to:
36 - \c std::istringstream
37 - \c std::stringstream
38 - \c std::wistringstream
39 - \c std::wstringstream
45 \tparam StreamType Class derived from \c std::basic_istream.
48 template <typename StreamType>
49 class BasicIStreamWrapper {
51 typedef typename StreamType::char_type Ch;
55 \param stream stream opened for read.
57 BasicIStreamWrapper(StreamType &stream) : stream_(stream), buffer_(peekBuffer_), bufferSize_(4), bufferLast_(0), current_(buffer_), readCount_(0), count_(0), eof_(false) {
63 \param stream stream opened for read.
64 \param buffer user-supplied buffer.
65 \param bufferSize size of buffer in bytes. Must >=4 bytes.
67 BasicIStreamWrapper(StreamType &stream, char* buffer, size_t bufferSize) : stream_(stream), buffer_(buffer), bufferSize_(bufferSize), bufferLast_(0), current_(buffer_), readCount_(0), count_(0), eof_(false) {
68 RAPIDJSON_ASSERT(bufferSize >= 4);
72 Ch Peek() const { return *current_; }
73 Ch Take() { Ch c = *current_; Read(); return c; }
74 size_t Tell() const { return count_ + static_cast<size_t>(current_ - buffer_); }
77 void Put(Ch) { RAPIDJSON_ASSERT(false); }
78 void Flush() { RAPIDJSON_ASSERT(false); }
79 Ch* PutBegin() { RAPIDJSON_ASSERT(false); return 0; }
80 size_t PutEnd(Ch*) { RAPIDJSON_ASSERT(false); return 0; }
82 // For encoding detection only.
83 const Ch* Peek4() const {
84 return (current_ + 4 - !eof_ <= bufferLast_) ? current_ : 0;
88 BasicIStreamWrapper();
89 BasicIStreamWrapper(const BasicIStreamWrapper&);
90 BasicIStreamWrapper& operator=(const BasicIStreamWrapper&);
93 if (current_ < bufferLast_)
97 readCount_ = bufferSize_;
98 bufferLast_ = buffer_ + readCount_ - 1;
101 if (!stream_.read(buffer_, static_cast<std::streamsize>(bufferSize_))) {
102 readCount_ = static_cast<size_t>(stream_.gcount());
103 *(bufferLast_ = buffer_ + readCount_) = '\0';
110 Ch peekBuffer_[4], *buffer_;
115 size_t count_; //!< Number of characters read
119 typedef BasicIStreamWrapper<std::istream> IStreamWrapper;
120 typedef BasicIStreamWrapper<std::wistream> WIStreamWrapper;
122 #if defined(__clang__) || defined(_MSC_VER)
126 RAPIDJSON_NAMESPACE_END
128 #endif // RAPIDJSON_ISTREAMWRAPPER_H_